home
***
CD-ROM
|
disk
|
FTP
|
other
***
search
/
Nebula 1
/
Nebula One.iso
/
Mail
/
pine3.92
/
pine
/
signals.c
< prev
next >
Wrap
C/C++ Source or Header
|
1996-03-14
|
26KB
|
991 lines
#if !defined(lint) && !defined(DOS)
static char rcsid[] = "$Id: signals.c,v 4.93 1996/03/15 07:13:42 hubert Exp $";
#endif
/*----------------------------------------------------------------------
T H E P I N E M A I L S Y S T E M
Laurence Lundblade and Mike Seibel
Networks and Distributed Computing
Computing and Communications
University of Washington
Administration Builiding, AG-44
Seattle, Washington, 98195, USA
Internet: lgl@CAC.Washington.EDU
mikes@CAC.Washington.EDU
Please address all bugs and comments to "pine-bugs@cac.washington.edu"
Pine and Pico are registered trademarks of the University of Washington.
No commercial use of these trademarks may be made without prior written
permission of the University of Washington.
Pine, Pico, and Pilot software and its included text are Copyright
1989-1996 by the University of Washington.
The full text of our legal notices is contained in the file called
CPYRIGHT, included with this distribution.
Pine is in part based on The Elm Mail System:
***********************************************************************
* The Elm Mail System - Revision: 2.13 *
* *
* Copyright (c) 1986, 1987 Dave Taylor *
* Copyright (c) 1988, 1989 USENET Community Trust *
***********************************************************************
----------------------------------------------------------------------*/
/*======================================================================
signals.c
Different signal handlers for different signals
- Catches all the abort signals, cleans up tty modes and then coredumps
- Not much to do for SIGHUP
- Not much to do for SIGTERM
- turn SIGWINCH into a KEY_RESIZE command
- No signals for ^Z/suspend, but do it here anyway
- Also set up the signal handlers, and hold signals for
critical imap sections of code.
====*/
#include "headers.h"
/*
* call used by TERM and HUP handlers to quickly close streams
*/
void fast_clean_up PROTO(());
void suspend_notice PROTO((char *));
void suspend_warning PROTO(());
#if defined(DOS) || defined(OS2)
#define SIG_PROTO(args) args
#else
#define SIG_PROTO(args) ()
#endif
/* SigType is defined in os.h and is either int or void */
static SigType auger_in_signal SIG_PROTO((int));
static SigType winch_signal SIG_PROTO((int));
static SigType usr2_signal SIG_PROTO((int));
static SigType alarm_signal SIG_PROTO((int));
static SigType intr_signal SIG_PROTO((int));
/*----------------------------------------------------------------------
Install handlers for all the signals we care to catch
----------------------------------------------------------------------*/
void
init_signals()
{
dprint(9, (debugfile, "init_signals()\n"));
#ifdef _WINDOWS
/* Only one signal works. */
signal(SIGALRM, (void *)alarm_signal);
signal(SIGHUP, (void *)hup_signal);
#else
#ifdef OS2
dont_interrupt();
signal(SIGALRM, (void *)alarm_signal);
signal(SIGHUP, (void *)hup_signal);
#else
#if defined(DOS)
dont_interrupt();
#else
#ifdef DEBUG
#define CUSHION_SIG (debug < 7)
#else
#define CUSHION_SIG (1)
#endif
if(CUSHION_SIG){
signal(SIGILL, auger_in_signal);
signal(SIGTRAP, auger_in_signal);
signal(SIGEMT, auger_in_signal);
signal(SIGBUS, auger_in_signal);
signal(SIGSEGV, auger_in_signal);
signal(SIGSYS, auger_in_signal);
signal(SIGQUIT, auger_in_signal);
/* Don't catch SIGFPE cause it's rare and we use it in a hack below*/
}
init_sigwinch();
signal(SIGHUP, hup_signal);
signal(SIGTERM, term_signal);
/*
* Set up SIGUSR2 to catch signal from other software using the
* c-client to tell us that other access to the folder is being
* attempted. THIS IS A TEST: if it turns out that simply
* going R/O when another pine is started or the same folder is opened,
* then we may want to install a smarter handler that uses idle time
* or even prompts the user to see if it's ok to give up R/O access...
*/
signal(SIGUSR2, usr2_signal);
#ifdef SA_RESTART
{
struct sigaction sa;
sa.sa_handler = alarm_signal;
memset(&sa.sa_mask, 0, sizeof(sa.sa_mask));
sa.sa_flags = SA_RESTART;
sigaction(SIGALRM, &sa, NULL);
}
#else
signal(SIGALRM, alarm_signal);
#endif
signal(SIGPIPE, SIG_IGN);
signal(SIGINT, SIG_IGN);
#ifdef SIGTSTP
/* Some unexplained behaviour on Ultrix 4.2 (Hardy) seems to be
resulting in Pine getting sent a SIGTSTP. Ignore it here.
probably better to ignore it than let it happen in any case
*/
signal(SIGTSTP, SIG_IGN);
#endif /* SIGTSTP */
#endif /* !DOS */
#endif /* !OS2 */
#endif /* !_WINDOWS */
}
/*----------------------------------------------------------------------
Return all signal handling back to normal
----------------------------------------------------------------------*/
void
end_signals(blockem)
int blockem;
{
#ifdef _WINDOWS
signal(SIGALRM, blockem ? SIG_IGN : SIG_DFL);
signal(SIGHUP, blockem ? SIG_IGN : SIG_DFL);
#else
#ifdef OS2
interrupt_ok();
signal(SIGALRM, blockem ? SIG_IGN : SIG_DFL);
signal(SIGHUP, blockem ? SIG_IGN : SIG_DFL);
#else
#ifdef DOS
interrupt_ok();
#else
#ifndef SIG_ERR
#define SIG_ERR (SigType (*)())-1
#endif
dprint(5, (debugfile, "end_signals(%d)\n", blockem));
if(signal(SIGILL, blockem ? SIG_IGN : SIG_DFL) == SIG_ERR){
fprintf(stderr, "Error resetting signals: %s\n",
error_description(errno));
exit(-1);
}
signal(SIGTRAP, blockem ? SIG_IGN : SIG_DFL);
signal(SIGEMT, blockem ? SIG_IGN : SIG_DFL);
signal(SIGBUS, blockem ? SIG_IGN : SIG_DFL);
signal(SIGSEGV, blockem ? SIG_IGN : SIG_DFL);
signal(SIGSYS, blockem ? SIG_IGN : SIG_DFL);
#ifdef RESIZING
signal(SIGWINCH, blockem ? SIG_IGN : SIG_DFL);
#endif
signal(SIGQUIT, blockem ? SIG_IGN : SIG_DFL);
#ifdef SIGTSTP
signal(SIGTSTP, blockem ? SIG_IGN : SIG_DFL);
#endif /* SIGTSTP */
signal(SIGHUP, blockem ? SIG_IGN : SIG_DFL);
signal(SIGALRM, blockem ? SIG_IGN : SIG_DFL);
signal(SIGTERM, blockem ? SIG_IGN : SIG_DFL);
signal(SIGINT, blockem ? SIG_IGN : SIG_DFL);
#endif /* !DOS */
#endif /* !OS2 */
#endif /* !_WINDOWS */
}
/*----------------------------------------------------------------------
Handle signals caused by aborts -- SIGSEGV, SIGILL, etc
Call panic which cleans up tty modes and then core dumps
----------------------------------------------------------------------*/
static SigType
auger_in_signal SIG_PROTO ((int sig))
{
end_signals(1); /* don't catch any more signals */
dprint(5, (debugfile, "auger_in_signal()\n"));
panic("Received abort signal"); /* clean up and get out */
exit(-1); /* in case panic doesn't kill us */
}
/*----------------------------------------------------------------------
handle hang up signal -- SIGHUP
Not much to do. Rely on periodic mail file check pointing.
----------------------------------------------------------------------*/
SigType
hup_signal()
{
#if !defined(DOS) || defined(_WINDOWS)
end_signals(1); /* don't catch any more signals */
dprint(1, (debugfile, "\n\n** Received SIGHUP **\n\n\n\n"));
fast_clean_up();
printf("\n\nPine finished. Received hang up signal\n\n");
#endif /* !DOS */
exit(0);
}
/*----------------------------------------------------------------------
handle terminate signal -- SIGTERM
Not much to do. Rely on periodic mail file check pointing.
----------------------------------------------------------------------*/
SigType
term_signal()
{
#if !defined(DOS) && !defined(OS2)
end_signals(1); /* don't catch any more signals */
dprint(1, (debugfile, "\n\n** Received SIGTERM **\n\n\n\n"));
fast_clean_up();
printf("\n\nPine finished. Received terminate signal\n\n");
#endif /* !DOS */
exit(0);
}
/*----------------------------------------------------------------------
Handle cleaning up mail streams and tty modes...
Not much to do. Rely on periodic mail file check pointing. Don't try
cleaning up screen or flushing output since stdout is likely already
gone. To be safe, though, we'll at least restore the original tty mode.
----------------------------------------------------------------------*/
void
fast_clean_up()
{
#if !defined(DOS) && !defined(OS2)
if(ps_global->inbox_stream != NULL && !ps_global->inbox_stream->lock){
if(ps_global->inbox_stream == ps_global->mail_stream)
ps_global->mail_stream = NULL;
mail_close(ps_global->inbox_stream);
}
if(ps_global->mail_stream != NULL &&
ps_global->mail_stream != ps_global->inbox_stream &&
!ps_global->mail_stream->lock)
mail_close(ps_global->mail_stream);
Raw(0);
#endif /* !DOS */
#if defined(DEBUG) && (!defined(DOS) || defined(_WINDOWS))
if(debugfile)
fclose(debugfile);
#endif
}
#if !defined(DOS) && !defined(OS2)
/*----------------------------------------------------------------------
handle hang up signal -- SIGUSR2
Not much to do. Rely on periodic mail file check pointing.
----------------------------------------------------------------------*/
static SigType
usr2_signal SIG_PROTO((int sig))
{
char c;
dprint(1, (debugfile, "\n\n** Received SIGUSR2 **\n\n\n\n"));
if(ps_global->inbox_stream
&& !ps_global->inbox_stream->lock
&& !ps_global->inbox_stream->rdonly
&& (c = *ps_global->inbox_stream->mailbox) != '{' && c != '*'){
mail_check(ps_global->inbox_stream); /* write latest state */
ps_global->inbox_stream->rdonly = 1; /* and become read-only */
mail_ping(ps_global->inbox_stream);
q_status_message(SM_ASYNC, 3, 7,
"Another Pine is accessing Inbox. Session now Read-Only.");
dprint(1, (debugfile, "** INBOX went read-only **\n\n"));
}
if(ps_global->mail_stream
&& !ps_global->mail_stream->lock
&& !ps_global->mail_stream->rdonly
&& (c = *ps_global->mail_stream->mailbox) != '{' && c != '*'){
mail_check(ps_global->mail_stream); /* write latest state */
ps_global->mail_stream->rdonly = 1; /* and become read-only */
mail_ping(ps_global->mail_stream);
q_status_message(SM_ASYNC, 3, 7,
"Another Pine is accessing folder. Session now Read-Only.");
dprint(1, (debugfile, "** secondary folder went read-only **\n\n"));
}
}
#endif
/*----------------------------------------------------------------------
Install signal handler to deal with window resize signal -- SIGWINCH
----------------------------------------------------------------------*/
void
init_sigwinch ()
{
#ifdef RESIZING
signal(SIGWINCH, winch_signal);
#endif
}
#ifdef RESIZING
/*----------------------------------------------------------------------
Handle window resize signal -- SIGWINCH
The planned strategy is just force a redraw command. This is similar
to new mail handling which forces a noop command. The signals are
help until pine reads input. Then a KEY_RESIZE is forced into the command
stream .
Note that ready_for_winch is only non-zero inside the read_char function,
so the longjmp only ever happens there, and it is really just a jump
from lower down in the function up to the top of that function. Its
purpose is to return a KEY_RESIZE from read_char when interrupted
out of the select lower down in read_char.
----------------------------------------------------------------------*/
extern jmp_buf winch_state;
extern int ready_for_winch, winch_occured;
SigType
static winch_signal SIG_PROTO((int sig))
{
dprint(9,(debugfile, "SIGWINCH ready_for_winch: %d winch_occured:%d\n",
ready_for_winch, winch_occured));
clear_cursor_pos();
init_sigwinch();
if(ready_for_winch)
longjmp(winch_state, 1);
else
winch_occured = 1;
}
#endif
#ifdef SIGCHLD
/*----------------------------------------------------------------------
Handle child status change -- SIGCHLD
The strategy here is to install the handler when we spawn a child, and
to let the system tell us when the child's state has changed. In the
mean time, we can do whatever. Typically, "whatever" is spinning in a
loop alternating between sleep and new_mail calls intended to keep the
IMAP stream alive.
----------------------------------------------------------------------*/
extern short child_signalled, child_jump;
extern jmp_buf child_state;
SigType
child_signal()
{
dprint(9,(debugfile, "SIGCHLD raised\n"));
child_signalled = 1;
if(child_jump)
longjmp(child_state, 1);
}
#endif
#define MAX_BM 80 /* max length of busy message */
static unsigned alarm_increment;
static int dotcount;
static char busy_message[MAX_BM + 1];
static int busy_alarm_outstanding;
static int busy_len;
static int final_message;
static int callcount;
static percent_done_t percent_done_ptr;
static char *display_chars[] = {
"<\\> ",
"<|> ",
"</> ",
"<-> ",
" ",
"<-> "
};
#define DISPLAY_CHARS_ROWS 6
#define DISPLAY_CHARS_COLS 4
/*
* Turn on a busy alarm.
*
* seconds -- alarm fires every seconds seconds
* msg -- the busy message to print in status line
* pc_func -- if non-null, call this function to get the percent done,
* (an integer between 0 and 100). If null, append dots.
* init_msg -- if non-zero, force out an immediate status message instead
* of waiting for first alarm (except see comments in code)
*
* Returns: 0 If busy alarm was already set up before we got here
* 1 If busy alarm was not already set up.
*/
int
busy_alarm(seconds, msg, pc_func, init_msg)
unsigned seconds;
char *msg;
percent_done_t pc_func;
int init_msg;
{
int retval = 1;
dprint(9,(debugfile, "busy_alarm(%d, %s, %p, %d)\n",
seconds, msg ? msg : "Busy", pc_func, init_msg));
/*
* If we're already busy'ing, and we don't have something special,
* just leave it alone.
*/
if(busy_alarm_outstanding){
retval = 0;
if(!msg && !pc_func)
return(retval);
}
if(seconds){
busy_alarm_outstanding = 1;
alarm_increment = seconds;
dotcount = 0;
percent_done_ptr = pc_func;
callcount = 0;
if(msg){
strncpy(busy_message, msg, MAX_BM);
final_message = 1;
}
else{
strcpy(busy_message, "Busy");
final_message = 0;
}
busy_message[MAX_BM] = '\0';
busy_len = strlen(busy_message);
if(init_msg){
char progress[MAX_SCREEN_COLS+1];
int space_left, slots_used;
final_message = 1;
space_left = (ps_global->ttyo ? ps_global->ttyo->screen_cols
: 80) -
busy_len - 2; /* 2 is for [] */
slots_used = max(0, min(space_left-3, 10));
if(percent_done_ptr && slots_used >= 4){
sprintf(progress, "%s |%*s|", busy_message, slots_used, "");
q_status_message(SM_ORDER, 0, 1, progress);
}
else{
dotcount++;
sprintf(progress, "%s%*s", busy_message,
DISPLAY_CHARS_COLS + 1, "");
q_status_message(SM_ORDER, 0, 1, progress);
}
/*
* We use display_message so that the initial message will
* be forced out only if there is not a previous message
* currently being displayed that hasn't been displayed for
* its min display time yet. In that case, we don't want
* to force out the initial message.
*/
display_message('x');
}
#ifdef _WINDOWS
mswin_setcursor (MSWIN_CURSOR_BUSY);
#endif
fflush(stdout);
}
/* set alarm */
(void)alarm(seconds);
return(retval);
}
/*
* If final_message was set when busy_alarm was called:
* and message_pri = -1 -- no final message queued
* else final message queued with min equal to message_pri
*/
void
cancel_busy_alarm(message_pri)
int message_pri;
{
dprint(9,(debugfile, "cancel_busy_alarm(%d)\n", message_pri));
(void)alarm(0);
if(busy_alarm_outstanding){
int space_left, slots_used;
busy_alarm_outstanding = 0;
if(final_message && message_pri >= 0){
char progress[MAX_SCREEN_COLS+1];
space_left = (ps_global->ttyo ? ps_global->ttyo->screen_cols : 80) -
busy_len - 2; /* 2 is for [] */
slots_used = max(0, min(space_left-3, 10));
if(percent_done_ptr && slots_used >= 4){
int left, right;
right = (slots_used - 4)/2;
left = slots_used - 4 - right;
sprintf(progress, "%s |%*s100%%%*s|",
busy_message, left, "", right, "");
q_status_message(SM_ORDER,
message_pri>=2 ? max(message_pri,3) : 0,
message_pri+2, progress);
}
else{
sprintf(progress, "%s%*sDONE", busy_message,
DISPLAY_CHARS_COLS - 4 + 1, "");
q_status_message(SM_ORDER,
message_pri>=2 ? max(message_pri,3) : 0,
message_pri+2, progress);
}
}
else
mark_status_dirty();
}
}
/*
* suspend_busy_alarm - continue previously installed busy_alarm.
*/
void
suspend_busy_alarm()
{
dprint(9,(debugfile, "suspend_busy_alarm\n"));
if(busy_alarm_outstanding)
alarm(0);
}
/*
* resume_busy_alarm - continue previously installed busy_alarm.
*/
void
resume_busy_alarm()
{
dprint(9,(debugfile, "resume_busy_alarm\n"));
if(busy_alarm_outstanding)
(void)alarm(alarm_increment);
}
static SigType
alarm_signal SIG_PROTO((int sig))
{
int space_left, slots_used;
char dbuf[MAX_SCREEN_COLS+1];
dprint(9,(debugfile, "alarm_signal()\n"));
space_left = (ps_global->ttyo ? ps_global->ttyo->screen_cols : 80) -
busy_len - 2; /* 2 is for [] */
slots_used = max(0, min(space_left-3, 10));
if(percent_done_ptr && slots_used >= 4){
int completed, pd;
char *s;
pd = (*percent_done_ptr)();
pd = min(max(0, pd), 100);
completed = (pd * slots_used) / 100;
sprintf(dbuf, "%s |%s%s%*s|", busy_message,
completed > 1 ? repeat_char(completed-1, pd==100 ? ' ' : '-') : "",
(completed > 0 && pd != 100) ? ">" : "",
slots_used - completed, "");
if(slots_used == 10){
s = dbuf + strlen(dbuf) - 8;
if(pd < 10){
s++; s++;
*s++ = '0' + pd;
}
else if(pd < 100){
s++;
*s++ = '0' + pd / 10;
*s++ = '0' + pd % 10;
}
else{
*s++ = '1';
*s++ = '0';
*s++ = '0';
}
*s = '%';
}
}
else{
char b[DISPLAY_CHARS_COLS + 2];
int md = DISPLAY_CHARS_ROWS - 2;
int ind;
ind = (dotcount == 0) ? md :
(dotcount == 1) ? md + 1 : ((dotcount-2) % md);
if(space_left >= DISPLAY_CHARS_COLS + 1){
b[0] = SPACE;
strcpy(b+1, display_chars[ind]);
}
else if(space_left >= 2){
b[0] = '.';
b[1] = '.';
b[2] = '.';
b[space_left] = '\0';
}
else
b[0] = '\0';
sprintf(dbuf, "%s%s", busy_message, b);
}
status_message_write(dbuf, 1);
dotcount++;
fflush(stdout);
#ifdef SA_RESTART
{
struct sigaction sa;
sa.sa_handler = alarm_signal;
memset(&sa.sa_mask, 0, sizeof(sa.sa_mask));
sa.sa_flags = SA_RESTART;
sigaction(SIGALRM, &sa, NULL);
}
#else
#if !defined(DOS) || defined(_WINDOWS)
signal(SIGALRM, alarm_signal);
#endif
#endif
(void)alarm(alarm_increment);
}
#ifdef DOS
/*
* For systems that don't have an alarm() call.
*/
void
fake_alarm_blip()
{
static time_t last_alarm = -1;
time_t now;
if(busy_alarm_outstanding &&
++callcount % 5 == 0 &&
(long)((now=time((time_t *)0)) - last_alarm) >= (long)alarm_increment){
last_alarm = now;
/* manually execute code that an alarm would cause to execute */
alarm_signal(1);
}
}
#endif
/*
* Command interrupt support.
*/
static SigType
intr_signal SIG_PROTO((int sig))
{
ps_global->intr_pending = 1;
}
void
intr_allow()
{
if(signal(SIGINT, intr_signal) == intr_signal)
return; /* already installed */
intr_proc(1); /* turn on interrupt char */
}
void
intr_disallow()
{
if(signal(SIGINT, SIG_IGN) == SIG_IGN) /* already off! */
return;
#ifdef POSIX_SIGNALS
sigrelse(SIGINT); /* unblock signal after longjmp */
#endif
ps_global->intr_pending = 0;
intr_proc(0); /* turn off interrupt char */
}
void
intr_handling_on()
{
if(signal(SIGINT, intr_signal) == intr_signal)
return; /* already installed */
intr_proc(1);
draw_cancel_keymenu();
}
void
intr_handling_off()
{
if(signal(SIGINT, SIG_IGN) == SIG_IGN) /* already off! */
return;
ps_global->intr_pending = 0;
intr_proc(0);
blank_keymenu(ps_global->ttyo->screen_rows - 2, 0);
ps_global->mangled_footer = 1;
}
/*----------------------------------------------------------------------
Suspend Pine. Reset tty and suspend. Suspend is finished when this returns
Args: The pine structure
Result: Execution suspended for a while. Screen will need redrawing
after this is done.
Instead of the usual handling of ^Z by catching a signal, we actually read
the ^Z and then clean up the tty driver, then kill ourself to stop, and
pick up where we left off when execution resumes.
----------------------------------------------------------------------*/
int
do_suspend(pine)
struct pine *pine;
{
time_t now;
int result, isremote, retval;
int orig_cols, orig_rows;
#ifdef DOS
static char *shell = NULL;
#define STD_SHELL "COMMAND.COM"
#else
#ifdef OS2
static char *shell = NULL;
#define STD_SHELL "CMD.EXE"
#else
char *shell;
#endif
#endif
if(!have_job_control()){
bogus_command(ctrl('Z'), F_ON(F_USE_FK, pine) ? "F1" : "?");
return(NO_OP_COMMAND);
}
if(F_OFF(F_CAN_SUSPEND, pine)){
q_status_message(SM_ORDER | SM_DING, 3, 3,
"Pine suspension not enabled - see help text");
return(NO_OP_COMMAND);
}
#ifdef _WINDOWS
/* suspend what? */
#else
isremote = (ps_global->mail_stream && ps_global->mail_stream->mailbox
&& (*ps_global->mail_stream->mailbox == '{'
|| (*ps_global->mail_stream->mailbox == '*'
&& *(ps_global->mail_stream->mailbox + 1) == '{')));
now = time((time_t *)0);
dprint(1, (debugfile, "\n\n --- %s - SUSPEND ---- %s",
isremote ? "REMOTE" : "LOCAL", ctime(&now)));
EndInverse();
end_keyboard(F_ON(F_USE_FK,pine));
end_tty_driver(pine);
end_screen(NULL);
#if defined(DOS) || defined(OS2)
#ifdef OS2
interrupt_ok();
#endif
suspend_notice("exit");
if (!shell){
char *p;
if (!((shell = getenv("SHELL")) || (shell = getenv("COMSPEC"))))
shell = STD_SHELL;
shell = cpystr(shell); /* copy in free storage */
for(p = shell; p = strchr(p, '/'); p++)
*p = '\\';
}
result = system(shell);
#else
if(F_ON(F_SUSPEND_SPAWNS, ps_global)){
PIPE_S *syspipe;
if(syspipe = open_system_pipe(NULL, NULL, NULL, PIPE_USER|PIPE_RESET)){
suspend_notice("exit");
#ifndef SIGCHLD
if(isremote)
suspend_warning();
#endif
(void) close_system_pipe(&syspipe);
}
}
else{
suspend_notice("fg");
if(isremote)
suspend_warning();
stop_process();
}
#endif /* DOS */
now = time((time_t *)0);
dprint(1, (debugfile, "\n\n ---- RETURN FROM SUSPEND ---- %s",
ctime(&now)));
init_screen();
init_tty_driver(pine);
init_keyboard(F_ON(F_USE_FK,pine));
#ifdef OS2
dont_interrupt();
#endif
orig_cols = pine->ttyo->screen_cols;
orig_rows = pine->ttyo->screen_rows;
fix_windsize(pine);
#if defined(DOS) || defined(OS2)
if(orig_cols != pine->ttyo->screen_cols ||
orig_rows != pine->ttyo->screen_rows)
retval = KEY_RESIZE;
else
#endif
retval = ctrl('L');;
#if defined(DOS) || defined(OS2)
if(result == -1)
q_status_message1(SM_ORDER | SM_DING, 3, 4,
"Error loading \"%s\"", shell);
#endif
if(isremote && (char *)mail_ping(ps_global->mail_stream) == NULL)
q_status_message(SM_ORDER | SM_DING, 4, 9,
"Suspended for too long, IMAP connection broken");
return(retval);
#endif /* !_WINDOWS */
}
/*----------------------------------------------------------------------
----*/
void
suspend_notice(s)
char *s;
{
printf("\nPine suspended. Give the \"%s\" command to come back.\n", s);
fflush(stdout);
}
/*----------------------------------------------------------------------
----*/
void
suspend_warning()
{
puts("Warning: Your IMAP connection will be closed if Pine");
puts("is suspended for more than 30 minutes\n");
fflush(stdout);
}
/*----------------------------------------------------------------------
----*/
void
fix_windsize(pine)
struct pine *pine;
{
mark_keymenu_dirty();
mark_status_dirty();
mark_titlebar_dirty();
clear_cursor_pos();
#ifdef RESIZING
get_windsize(pine->ttyo);
#endif /* RESIZING */
}
#if defined(DOS) || defined(OS2)
SigType (*hold_int)(int), (*hold_term)(int), (*hold_quit)(int);
#else
SigType (*hold_hup)(), (*hold_int)(), (*hold_term)(), (*hold_usr2)(),
(*hold_quit)();
#endif
/*----------------------------------------------------------------------
Ignore signals when imap is running through critical code
Args: stream -- The stream on which critical operation is proceeding
----*/
void
mm_critical(stream)
MAILSTREAM *stream;
{
stream = stream; /* For compiler complaints that this isn't used */
#if !defined(DOS) && !defined(OS2)
hold_hup = signal(SIGHUP, SIG_IGN);
hold_usr2 = signal(SIGUSR2, SIG_IGN);
#endif
hold_int = signal(SIGINT, SIG_IGN);
hold_term = signal(SIGTERM, SIG_IGN);
dprint(9, (debugfile, "Done with IMAP critical on %s\n",
stream ? stream->mailbox : "<no folder>" ));
}
/*----------------------------------------------------------------------
Reset signals after critical imap code
----*/
void
mm_nocritical(stream)
MAILSTREAM *stream;
{
stream = stream; /* For compiler complaints that this isn't used */
#if !defined(DOS) && !defined(OS2)
(void)signal(SIGHUP, hold_hup);
(void)signal(SIGUSR2, hold_usr2);
#endif
(void)signal(SIGINT, hold_int);
(void)signal(SIGTERM, hold_term);
dprint(9, (debugfile, "Done with IMAP critical on %s\n",
stream ? stream->mailbox : "<no folder>" ));
}